from flask import current_app
from flask import g
from flask import render_template
from flask import request
from flask.json import jsonify

from info import constants, db
from info.models import News, Category, User
from info.utils.captcha.response_code import RET
from info.utils.commons import user_login_data
from info.utils.image_storage import image_storage
from . import profile_blu

#功能描述：作者新闻列表
#请求路径：/user/other_news_list
#请求方式：GET
#请求参数：p,user_id
#返回值：errno,errmsg
@profile_blu.route('/other_news_list')

def other_news_list():
    """
    思路分析：
    # 1.获取参数
    # 2.参数校验，为空判断，参数类型转换
    # 3.取出作者对象
    # 4.判断作者对象是否存在
    # 5.分页获取作者新闻列表
    # 6.获取分页对象属性，总页数，当前页数，当前页对象
    # 7.返回响应
    :return:
    """
    # 1.获取参数
    page=request.args.get("p",1)
    author_id=request.args.get("user_id")
    # 2.参数校验，为空判断，参数类型转换
    if not author_id:
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")
    try:
        page=int(page)
    except Exception as e:
        current_app.logger.error(e)
        page=1

    # 3.取出作者对象
    try:
        author=User.query.get(author_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")
    # 4.判断作者对象是否存在
    if not author:
        return jsonify(errno=RET.NODATA,errmsg="作者不存在")
    # 5.分页获取作者新闻列表
    try:
        paginate=author.news_list.order_by(News.create_time.desc()).paginate(page,1,False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="查询新闻失败")
    # 6.获取分页对象属性，总页数，当前页数，当前页对象
    total_page=paginate.pages
    current_page=paginate.page
    items=paginate.items
    news_list=[]
    for item in items:
        news_list.append(item.to_dict())

    # 7.返回响应
    data ={
        "total_page":total_page,
        "current_page":current_page,
        "news_list":news_list
    }
    return jsonify(errno=RET.OK,errmsg="操作成功",data=data)

#功能描述：作者详情界面
#请求路径：/user/other
#请求方式：GET
#请求参数：id
#返回值：渲染other.html页面，字典data数据
@profile_blu.route('/other')
@user_login_data
def other_info():
    """
    思路分析：
    # 1.获取作者编号
    # 2.判断编号是否存在
    # 3.查询作者对象
    # 4.判断作者对象是否存在
    # 5.拼接数据，渲染页面
    :return:
    """
    # 1.获取作者编号
    author_id=request.args.get('id')
    # 2.判断编号是否存在
    if not author_id:
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")
    # 3.查询作者对象
    try:
        author=User.query.get(author_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.NODATA,errmsg="查询用户失败")
    # 4.判断作者对象是否存在
    if not author:
        return jsonify(errno=RET.NODATA,errmsg="作者不存在")
    #4.2 判断当前用户是否关注了该作者：
    is_followed=False
    if g.user:
        if g.user in author.followers:
            is_followed=True
    # 5.拼接数据，渲染页面
    data={
        "author_info":author.to_dict() if author else "",
        "user_info":g.user.to_dict() if g.user else "",
        "is_followed":is_followed
    }
    return  render_template('news/other.html',data=data)

#功能描述：获取我的关注列表
#请求路径：/user/user_follow
#请求方式：GET
#请求参数：p
#返回值：渲染user_follow.html页面，字典data数据
@profile_blu.route('/user_follow')
@user_login_data
def user_follow():
    """
    思路分析：
    # 1.获取参数
    # 2.参数数据类型转换
    # 3.分页查询
    # 4.获取分页对象属性，总页数，当前页，对象列表
    # 5.对象列表转成字典列表
    # 6.拼接数据渲染页面

    :return:
    """
    # 1.获取参数
    page=request.args.get('p',1)
    # 2.参数数据类型转换
    try:
        page=int(page)
    except Exception as e:
        current_app.logger.error(e)
        page=1
    # 3.分页查询
    try:
        paginate=g.user.followed.paginate(page,4,False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取用户失败")
    # 4.获取分页对象属性，总页数，当前页，对象列表
    totalPage=paginate.pages
    currentPage=paginate.page
    items=paginate.items
    # 5.对象列表转成字典列表
    user_list=[]
    for item in items:
        user_list.append(item.to_dict())
    # 6.拼接数据渲染页面
    data ={
        "totalPage":totalPage,
        "currentPage":currentPage,
        "user_list":user_list

    }
    return render_template('news/user_follow.html',data=data)

#功能描述：新闻列表展示
#请求路径：/user/news_list
#请求方式：GET
#请求参数：P
#返回值：GET渲染user_news_list.html页面
@profile_blu.route('/news_list')
@user_login_data
def news_list():
    """
    思路分析：
    # 1.获取参数
    # 2.参数数据类型转换
    # 3.分页查询
    # 4.获取分页对象属性，总页数，当前页，对象列表
    # 5.对象列表转成字典列表
    # 6.拼接数据渲染页面
    :return:
    """
    # 1.获取参数
    page=request.args.get('p',1)
    # 2.参数数据类型转换
    try:
        page=int(page)
    except Exception as e:
        current_app.logger.error(e)
        page=1
    # 3.分页查询
    try:
        paginate=g.user.news_list.order_by(News.create_time.desc()).paginate(page,10,False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="新闻获取失败")
    # 4.获取分页对象属性，总页数，当前页，对象列表
    totalPage=paginate.pages
    currentPage=paginate.page
    items=paginate.items

    # 5.对象列表转成字典列表
    news_list=[]
    for item in items:
        news_list.append(item.to_review_dict())

    # 6.拼接数据渲染页面
    data={
        "totalPage":totalPage,
        "currentPage":currentPage,
        "news_list":news_list
    }
    return render_template("news/user_news_list.html",data=data)


#功能描述：获取/设置，新闻发布
#请求路径: /user/news_release
# 请求方式:GET,POST
# 请求参数:GET无, POST ,title, category_id,digest,index_image,content
# 返回值:GET请求,user_news_release.html, data分类列表字段数据, POST,errno,errmsg
@profile_blu.route('/news_release',methods=['GET','POST'])
@user_login_data
def news_release():
    """
    思路分析：
    # 1.第一次进来GET请求，展示页面
    # 2.获取参数
    # 3.校验参数
    # 4.上传图片
    # 5.判断图片是否上传成功
    # 6.创建新闻对象，设置属性
    # 7.保存新闻对象到数据库
    # 8.返回响应
    :return:
    """
    # 1.第一次进来GET请求，展示页面
    if request.method=="GET":
        try:
            categories=Category.query.all()
            categories.pop(0)
        except Exception as e:
            current_app.logger.error(e)
            return jsonify(errno=RET.DBERR,errmsg="分类查询失败")
        #转成字典列表
        category_list=[]
        for category in categories:
            category_list.append(category.to_dict())



        return render_template('news/user_news_release.html',categories=categories)

    # 2.获取参数
    title=request.form.get('title')
    category_id = request.form.get('category_id')
    digest=request.form.get('digest')
    index_image = request.files.get('index_image')
    content = request.form.get('content')
    # 3.校验参数
    if not all([title,category_id,digest,index_image,content]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")

    # 4.上传图片
    try:
        image_name=image_storage(index_image.read())
    except Exception as e:
        current_app.logger.error(e)
        return  jsonify(errno=RET.THIRDERR,errmsg="七牛云上传异常")
    # 5.判断图片是否上传成功
    if not image_name:
        return jsonify(errno=RET.NODATA,errmsg="图片上传失败")

    # 6.创建新闻对象，设置属性
    news=News()
    news.title=title
    news.source="个人发布"
    news.digest=digest
    news.content=content
    news.index_image_url=constants.QINIU_DOMIN_PREFIX +image_name
    news.category_id=category_id
    news.user_id=g.user.id
    news.status=1

    # 7.保存新闻对象到数据库
    try:
        db.session.add(news)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=RET.DBERR,errmsg="新闻发布失败")
    # 8.返回响应
    return jsonify(errno=RET.OK,errmsg="操作成功")
#功能描述：获取收藏列表
#请求路径：/user/collection
#请求方式：GET
#请求参数：P（页数）
#返回值：user_collection.html页面
@profile_blu.route('/collection')
@user_login_data
def news_collection():
    """
    思路分析：
    # 1.获取参数，获取默认第一页
    # 2.参数类型转换，字符串转整数
    # 3.分页查询
    # 4.获取到分页对象属性，总页数，当前页，当前页对象
    # 5.对象列表，转成字典列表
    # 6.拼接数据，渲染页面
    :return:
    """
    # 1.获取参数，获取默认第一页
    page=request.args.get('p',1)
    # 2.参数类型转换，字符串转整数
    try:
        page=int(page)
    except Exception as e:
        current_app.logger.error(e)
        page=1


    # 3.分页查询
    try:
        paginate=g.user.collection_news.order_by(News.create_time.desc()).paginate(page,3,False)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg="获取新闻失败")
    # 4.获取到分页对象属性，总页数，当前页，当前页对象
    totalPage=paginate.pages
    currentPage=paginate.page
    items=paginate.items
    # 5.对象列表，转成字典列表
    news_list=[]
    for item in items:
        news_list.append(item.to_dict())
    # 6.拼接数据，渲染页面
    data ={
        "totalPage":totalPage,
        "currentPage":currentPage,
        "news_list":news_list
    }
    return render_template('news/user_collection.html',data=data)





#功能描述：密码修改
#请求路径：/user/pass_info
#请求方式：GET，POST
#请求参数：GET无，POST有，old_password,new_password
#返回值：GET请求，user_pass_info,html页面，data字典数据，POST请求：errno,errmsg
@profile_blu.route('/pass_info',methods=['GET','POST'])
@user_login_data
def pass_info():
    """
    思路分析：
    1.第一次进来GET，渲染页面
    2.获取参数
    3.校验参数
    4.判断旧密码是否正确
    5.设置新密码
    6.返回响应
    :return:
    """
    # 1.第一次进来GET，渲染页面
    if request.method=="GET":
        return render_template('news/user_pass_info.html')

    # 2.获取参数
    old_password=request.json.get("old_password")
    new_password=request.json.get("new_password")
    # 3.校验参数
    if not all ([old_password,new_password]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")


    # 4.判断旧密码是否正确
    if not g.user.check_passowrd(old_password):
        return jsonify(errno=RET.DATAERR,errmsg="旧密码错误")
    # 5.设置新密码
    g.user.password=new_password
    # 6.返回响应
    return jsonify(errno=RET.OK,errmsg="修改成功")


#功能描述：图像设置
#请求路径：/user/pic_info
#请求方式：GET,POST
#请求参数：无，POST有参数，avatar
#返回值：GET请求：user_pic_info.html页面，data字典数据，POST请求：errno,errmsg,avatar_url
@profile_blu.route('/pic_info',methods=['GET','POST'])
@user_login_data
def pic_info():
    """
    思路分析：
    1.第一次进来GET请求，直接返回页面，携带用户数据
    2.获取参数
    3.校验参数
    4.上传头像
    5.判断头像是否上传成功
    6.设置用户头像属性
    7.返回响应携带头像
    :return:
    """
    # 1.第一次进来GET请求，直接返回页面，携带用户数据
    if request.method=="GET":
        data={
            'user_info':g.user.to_dict() if g.user else ""
        }
        return render_template("news/user_pic_info.html",data=data)
    # 2.获取参数
    avatar=request.files.get("avatar")
    # 3.校验参数
    if not avatar:
        return jsonify(errno=RET.PARAMERR,errmsg="图像不能为空")
    # 4.上传头像
    try:
        image_name=image_storage(avatar.read())
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.THIRDERR,errmsg="七牛云上传异常")

    # 5.判断头像是否上传成功
    if not image_name:
        return jsonify(errno=RET.NODATA,errmsg="图片上传失败")


    # 6.设置用户头像属性
    g.user.avatar_url=image_name
    # 7.返回响应携带头像
    data={
        "avatar_url":constants.QINIU_DOMIN_PREFIX + image_name
    }
    return jsonify(errno=RET.OK,errmsg="上传图片成功",data=data)


#功能描述：展示用户基本信息
#请求方式：GET，POST
#请求参数：POST请求
#返回值：errno,errmsg
@profile_blu.route('/base_info',methods=['GET','POST'])
@user_login_data
def base_info():
    """
    思路分析：
    1.如果第一次进入的是GET请求，携带用户数据渲染页面
    2.获取参数
    3.校验参数
    4.设置用户信息
    5.返回响应
    :return:
    """
    # 1.如果第一次进入的是GET请求，携带用户数据渲染页面
    if request.method=='GET':
        data={
            'user_info':g.user.to_dict() if g.user else ''

        }
        return render_template('news/user_base_info.html',data=data)

    # 2.获取参数
    nick_name=request.json.get("nick_name")
    signature=request.json.get("signature")
    gender=request.json.get("gender")
    # 3.校验参数
    if not all ([nick_name,signature,gender]):
        return jsonify(errno=RET.PARAMERR,errmsg="参数不全")
    if not gender in ['MAN','WOMAN']:
        return  jsonify(errno=RET.DATAERR,errmsg="性别异常")
    # 4.设置用户信息
    g.user.nick_name=nick_name
    g.user.signature=signature
    g.user.gender=gender
    # 5.返回响应
    return jsonify(errno=RET.OK,errmsg="操作成功")



#展示个人中心
@profile_blu.route('/show_user_info')
@user_login_data
def show_user_info():
    data={
        'user_info':g.user.to_dict() if g.user else ''
    }

    return render_template('news/user.html',data=data)